1

我们学习过将配置信息通过@Value()的方法注入到对象的变量中。这是由于对象是由spring统一托管的(保证了单例模式)。那于对于非spring托管的类,如果注入注入数据呢?

比如:我们想把配置信息的值,注入到类的静态变量中。

错误做法

application.properties

spring.redis.host=test
@Component
public class RedisServiceImpl implements RedisService {
    ...
    @Value("${spring.redis.host}")
    static public String host;
    
    @Value("${spring.redis.port}")
    static public Integer port;
    ...
    
     static public JedisPool getJedisPool() {
        if (RedisServiceImpl.host == null) {
            logger.info("host 未注入");
        }
    }

控制台打印为: "host 未注入

正确的方法

@Component
public class RedisServiceImpl implements RedisService {
    ...
    static public String host;
    static public Integer port;
    
    @Value("${spring.redis.host}")
    public void setHost(String host) {
        RedisServiceImpl.host = host;
    }

    @Value("${spring.redis.port}")
    public void setPort(Integer port) {
        RedisServiceImpl.port = port;
    }
    ...
    
     static public JedisPool getJedisPool() {
        if (RedisServiceImpl.host == null) {
            logger.info("host 未注入");
        } else {
            logger.info("host 值为" + RedisServiceImpl.host);
        }
    }

控制台正确的打印了注入的值。

原因猜想

spring在启动时会进行组件扫描,打描到RedisServiceImpl时,发现其类使用了@Component注解。于是,初始化对象 RedisServiceImpl。初始化过程中,对方法进行扫描,当扫描到使用@Value注解的方法时,调用方法,并注入需要注入的值。

然后:我们使用了一个小的技巧: 在这个自动执行的方法中,将值设置给了类。进而实现了,在启动时将值注入到类的目标。

其实:spring不对类进行托管,所以也就不可能将值注入到类。所以我们的上述方法,应该描述为:将值注入给方法,然后在方法中,使用传入的值为类进行数据初始化。

这时,我们也就清楚为什么使用@Value()注解时,无法将值注入的原因了:
如果将@Value(),直接加到静态私有变量上,则在初始化对象时,由于静态私有变量属于类,所以spring未对类进行操作 -- 错误。


潘杰
3.1k 声望239 粉丝